home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 15 / CU Amiga Magazine's Super CD-ROM 15 (1997)(EMAP Images)(GB)[!][issue 1997-10].iso / CUCD / Graphics / Ghostscript / source / gdevtfax.c < prev    next >
C/C++ Source or Header  |  1997-03-21  |  13KB  |  425 lines

  1. /* Copyright (C) 1994, 1995, 1997 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* gdevtfax.c */
  20. /* TIFF and fax devices */
  21. #include "gdevprn.h"
  22. #include "gdevtifs.h"            /* for TIFF output only */
  23. #include "strimpl.h"
  24. #include "scfx.h"
  25.  
  26. /* Define the device parameters. */
  27. #define X_DPI 204
  28. #define Y_DPI 196
  29. #define LINE_SIZE ((X_DPI * 101 / 10 + 7) / 8)    /* max bytes per line */
  30.  
  31. /* The device descriptors */
  32.  
  33. dev_proc_open_device(gdev_fax_open);
  34. private dev_proc_print_page(faxg3_print_page);
  35. private dev_proc_print_page(faxg32d_print_page);
  36. private dev_proc_print_page(faxg4_print_page);
  37. private dev_proc_print_page(tiffcrle_print_page);
  38. private dev_proc_print_page(tiffg3_print_page);
  39. private dev_proc_print_page(tiffg32d_print_page);
  40. private dev_proc_print_page(tiffg4_print_page);
  41.  
  42. struct gx_device_tfax_s {
  43.     gx_device_common;
  44.     gx_prn_device_common;
  45.     gdev_tiff_state tiff;        /* for TIFF output only */
  46. };
  47. typedef struct gx_device_tfax_s gx_device_tfax;
  48.  
  49. #define tfdev ((gx_device_tfax *)dev)
  50.  
  51. /* Define procedures that adjust the paper size. */
  52. private gx_device_procs gdev_fax_std_procs =
  53.   prn_procs(gdev_fax_open, gdev_prn_output_page, gdev_prn_close);
  54.  
  55. gx_device_tfax far_data gs_faxg3_device =
  56. {   prn_device_std_body(gx_device_tfax, gdev_fax_std_procs, "faxg3",
  57.     DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  58.     X_DPI, Y_DPI,
  59.     0,0,0,0,            /* margins */
  60.     1, faxg3_print_page)
  61. };
  62.  
  63. gx_device_tfax far_data gs_faxg32d_device =
  64. {   prn_device_std_body(gx_device_tfax, gdev_fax_std_procs, "faxg32d",
  65.     DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  66.     X_DPI, Y_DPI,
  67.     0,0,0,0,            /* margins */
  68.     1, faxg32d_print_page)
  69. };
  70.  
  71. gx_device_tfax far_data gs_faxg4_device =
  72. {   prn_device_std_body(gx_device_tfax, gdev_fax_std_procs, "faxg4",
  73.     DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  74.     X_DPI, Y_DPI,
  75.     0,0,0,0,            /* margins */
  76.     1, faxg4_print_page)
  77. };
  78.  
  79. gx_device_tfax far_data gs_tiffcrle_device =
  80. {   prn_device_std_body(gx_device_tfax, gdev_fax_std_procs, "tiffcrle",
  81.     DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  82.     X_DPI, Y_DPI,
  83.     0,0,0,0,            /* margins */
  84.     1, tiffcrle_print_page)
  85. };
  86.  
  87. gx_device_tfax far_data gs_tiffg3_device =
  88. {   prn_device_std_body(gx_device_tfax, gdev_fax_std_procs, "tiffg3",
  89.     DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  90.     X_DPI, Y_DPI,
  91.     0,0,0,0,            /* margins */
  92.     1, tiffg3_print_page)
  93. };
  94.  
  95. gx_device_tfax far_data gs_tiffg32d_device =
  96. {   prn_device_std_body(gx_device_tfax, gdev_fax_std_procs, "tiffg32d",
  97.     DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  98.     X_DPI, Y_DPI,
  99.     0,0,0,0,            /* margins */
  100.     1, tiffg32d_print_page)
  101. };
  102.  
  103. gx_device_tfax far_data gs_tiffg4_device =
  104. {   prn_device_std_body(gx_device_tfax, gdev_fax_std_procs, "tiffg4",
  105.     DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  106.     X_DPI, Y_DPI,
  107.     0,0,0,0,            /* margins */
  108.     1, tiffg4_print_page)
  109. };
  110.  
  111. /* Open the device, adjusting the paper size. */
  112. int
  113. gdev_fax_open(gx_device *dev)
  114. {    if ( dev->width >= 1680 && dev->width <= 1736 )
  115.       {    /* Adjust width for A4 paper. */
  116.         dev->width = 1728;
  117.       }
  118.     else if ( dev->width >= 2000 && dev->width <= 2056 )
  119.       {    /* Adjust width for B4 paper. */
  120.         dev->width = 2048;
  121.       }
  122.     return gdev_prn_open(dev);
  123. }
  124.  
  125. /* Initialize the stream state with a set of default parameters. */
  126. /* These select the same defaults as the CCITTFaxEncode filter, */
  127. /* except we set BlackIs1 = true. */
  128. void
  129. gdev_fax_init_state(stream_CFE_state *ss, const gx_device_printer *pdev)
  130. {    (*s_CFE_template.set_defaults)((stream_state *)ss);
  131.     ss->Columns = pdev->width;
  132.     ss->Rows = pdev->height;
  133.     ss->BlackIs1 = true;
  134. }
  135.  
  136. /* Send the page to the printer. */
  137. int
  138. gdev_stream_print_page(gx_device_printer *pdev, FILE *prn_stream,
  139.   const stream_template _ds *temp, stream_state *ss)
  140. {    gs_memory_t *mem = &gs_memory_default;
  141.     int code;
  142.     stream_cursor_read r;
  143.     stream_cursor_write w;
  144.     int in_size = gdev_mem_bytes_per_scan_line((gx_device *)pdev);
  145.     int lnum;
  146.     byte *in;
  147.     byte *out;
  148.     /* If the file is 'nul', don't even do the writes. */
  149.     bool nul = !strcmp(pdev->fname, "nul");
  150.  
  151.     /* Initialize the common part of the encoder state. */
  152.     ss->template = temp;
  153.     ss->memory = mem;
  154.     /* Now initialize the encoder. */
  155.     code = (*temp->init)(ss);
  156.     if ( code < 0 )
  157.       return_error(gs_error_limitcheck); /* bogus, but as good as any */
  158.  
  159.     /* Allocate the buffers. */
  160.     in = gs_alloc_bytes(mem, temp->min_in_size + in_size + 1, "gdev_stream_print_page(in)");
  161. #define out_size 1000
  162.     out = gs_alloc_bytes(mem, out_size, "gdev_stream_print_page(out)");
  163.     if ( in == 0 || out == 0 )
  164.     {    code = gs_note_error(gs_error_VMerror);
  165.         goto done;
  166.     }
  167.  
  168.     /* Set up the processing loop. */
  169.     lnum = 0;
  170.     r.ptr = r.limit = in - 1;
  171.     w.ptr = out - 1;
  172.     w.limit = w.ptr + out_size;
  173.  
  174.     /* Process the image. */
  175.     for ( ; ; )
  176.     {    int status;
  177.         if_debug7('w', "[w]lnum=%d r=0x%lx,0x%lx,0x%lx w=0x%lx,0x%lx,0x%lx\n", lnum,
  178.               (ulong)in, (ulong)r.ptr, (ulong)r.limit,
  179.               (ulong)out, (ulong)w.ptr, (ulong)w.limit);
  180.         status = (*temp->process)(ss, &r, &w, lnum == pdev->height);
  181.         if_debug7('w', "...%d, r=0x%lx,0x%lx,0x%lx w=0x%lx,0x%lx,0x%lx\n", status,
  182.               (ulong)in, (ulong)r.ptr, (ulong)r.limit,
  183.               (ulong)out, (ulong)w.ptr, (ulong)w.limit);
  184.         switch ( status )
  185.         {
  186.         case 0:            /* need more input data */
  187.         {    uint left;
  188.             if ( lnum == pdev->height )
  189.               goto ok;
  190.             left = r.limit - r.ptr;
  191.             memcpy(in, r.ptr + 1, left);
  192.             gdev_prn_copy_scan_lines(pdev, lnum++, in + left, in_size);
  193.             r.limit = in + left + in_size - 1;
  194.             r.ptr = in - 1;
  195.         }    break;
  196.         case 1:            /* need to write output */
  197.             if ( !nul )
  198.               fwrite(out, 1, w.ptr + 1 - out, prn_stream);
  199.             w.ptr = out - 1;
  200.             break;
  201.         }
  202.     }
  203.  
  204. ok:
  205.     /* Write out any remaining output. */
  206.     if ( !nul )
  207.       fwrite(out, 1, w.ptr + 1 - out, prn_stream);
  208.  
  209. done:
  210.     gs_free_object(mem, out, "gdev_stream_print_page(out)");
  211.     gs_free_object(mem, in, "gdev_stream_print_page(in)");
  212.     if ( temp->release != 0 )
  213.       (*temp->release)(ss);
  214.     return code;
  215. }
  216. /* Print a fax page.  Other fax drivers use this. */
  217. int
  218. gdev_fax_print_page(gx_device_printer *pdev, FILE *prn_stream,
  219.   stream_CFE_state *ss)
  220. {    return gdev_stream_print_page(pdev, prn_stream, &s_CFE_template,
  221.                       (stream_state *)ss);
  222. }
  223.  
  224. /* Print a 1-D Group 3 page. */
  225. private int
  226. faxg3_print_page(gx_device_printer *pdev, FILE *prn_stream)
  227. {    stream_CFE_state state;
  228.     gdev_fax_init_state(&state, pdev);
  229.     state.EndOfLine = true;
  230.     state.EndOfBlock = false;
  231.     return gdev_fax_print_page(pdev, prn_stream, &state);
  232. }
  233.  
  234. /* Print a 2-D Group 3 page. */
  235. private int
  236. faxg32d_print_page(gx_device_printer *pdev, FILE *prn_stream)
  237. {    stream_CFE_state state;
  238.     gdev_fax_init_state(&state, pdev);
  239.     state.K = (pdev->y_pixels_per_inch < 100 ? 2 : 4);
  240.     state.EndOfLine = true;
  241.     state.EndOfBlock = false;
  242.     return gdev_fax_print_page(pdev, prn_stream, &state);
  243. }
  244.  
  245. /* Print a Group 4 page. */
  246. private int
  247. faxg4_print_page(gx_device_printer *pdev, FILE *prn_stream)
  248. {    stream_CFE_state state;
  249.     gdev_fax_init_state(&state, pdev);
  250.     state.K = -1;
  251.     state.EndOfBlock = false;
  252.     return gdev_fax_print_page(pdev, prn_stream, &state);
  253. }
  254.  
  255. /* ---------------- TIFF output ---------------- */
  256.  
  257. #include "slzwx.h"
  258. #include "srlx.h"
  259.  
  260. /* Device descriptors for TIFF formats other than fax. */
  261. private dev_proc_print_page(tifflzw_print_page);
  262. private dev_proc_print_page(tiffpack_print_page);
  263.  
  264. gx_device_tfax far_data gs_tifflzw_device =
  265. {   prn_device_std_body(gx_device_tfax, prn_std_procs, "tifflzw",
  266.     DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  267.     X_DPI, Y_DPI,
  268.     0,0,0,0,            /* margins */
  269.     1, tifflzw_print_page)
  270. };
  271.  
  272. gx_device_tfax far_data gs_tiffpack_device =
  273. {   prn_device_std_body(gx_device_tfax, prn_std_procs, "tiffpack",
  274.     DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  275.     X_DPI, Y_DPI,
  276.     0,0,0,0,            /* margins */
  277.     1, tiffpack_print_page)
  278. };
  279.  
  280. /* Define the TIFF directory we use, beyond the standard entries. */
  281. /* NB: this array is sorted by tag number (assumed below) */
  282. typedef struct tiff_mono_directory_s {
  283.     TIFF_dir_entry    BitsPerSample;
  284.     TIFF_dir_entry    Compression;
  285.     TIFF_dir_entry    Photometric;
  286.     TIFF_dir_entry    FillOrder;
  287.     TIFF_dir_entry    SamplesPerPixel;
  288.     TIFF_dir_entry    T4T6Options;
  289.     TIFF_dir_entry    CleanFaxData;
  290. } tiff_mono_directory;
  291. private const tiff_mono_directory far_data dir_mono_template = {
  292.     { TIFFTAG_BitsPerSample,    TIFF_SHORT, 1, 1 },
  293.     { TIFFTAG_Compression,    TIFF_SHORT, 1, Compression_CCITT_T4 },
  294.     { TIFFTAG_Photometric,    TIFF_SHORT, 1, Photometric_min_is_white },
  295.     { TIFFTAG_FillOrder,    TIFF_SHORT, 1, FillOrder_LSB2MSB },
  296.     { TIFFTAG_SamplesPerPixel,    TIFF_SHORT, 1, 1 },
  297.     { TIFFTAG_T4Options,    TIFF_LONG,  1, 0 },
  298.     { TIFFTAG_CleanFaxData,    TIFF_SHORT, 1, CleanFaxData_clean },
  299. };
  300.  
  301. /* Forward references */
  302. private int tfax_begin_page(P3(gx_device_tfax *, FILE *, const tiff_mono_directory *));
  303.  
  304. /* Print a fax-encoded page. */
  305. private int
  306. tifff_print_page(gx_device_printer *dev, FILE *prn_stream,
  307.   stream_CFE_state *pstate, tiff_mono_directory *pdir)
  308. {    int code;
  309.  
  310.     tfax_begin_page(tfdev, prn_stream, pdir);
  311.     pstate->FirstBitLowOrder = true;    /* decoders prefer this */
  312.     code = gdev_fax_print_page(dev, prn_stream, pstate);
  313.     gdev_tiff_end_page(&tfdev->tiff, prn_stream);
  314.     return code;
  315. }
  316. private int
  317. tiffcrle_print_page(gx_device_printer *dev, FILE *prn_stream)
  318. {    stream_CFE_state state;
  319.     tiff_mono_directory dir;
  320.  
  321.     gdev_fax_init_state(&state, dev);
  322.     state.EndOfLine = false;
  323.     state.EncodedByteAlign = true;
  324.     dir = dir_mono_template;
  325.     dir.Compression.value = Compression_CCITT_RLE;
  326.     dir.T4T6Options.tag = TIFFTAG_T4Options;
  327.     dir.T4T6Options.value = T4Options_fill_bits;
  328.     return tifff_print_page(dev, prn_stream, &state, &dir);
  329. }
  330. private int
  331. tiffg3_print_page(gx_device_printer *dev, FILE *prn_stream)
  332. {    stream_CFE_state state;
  333.     tiff_mono_directory dir;
  334.  
  335.     gdev_fax_init_state(&state, dev);
  336.     state.EndOfLine = true;
  337.     state.EncodedByteAlign = true;
  338.     dir = dir_mono_template;
  339.     dir.Compression.value = Compression_CCITT_T4;
  340.     dir.T4T6Options.tag = TIFFTAG_T4Options;
  341.     dir.T4T6Options.value = T4Options_fill_bits;
  342.     return tifff_print_page(dev, prn_stream, &state, &dir);
  343. }
  344. private int
  345. tiffg32d_print_page(gx_device_printer *dev, FILE *prn_stream)
  346. {    stream_CFE_state state;
  347.     tiff_mono_directory dir;
  348.  
  349.     gdev_fax_init_state(&state, dev);
  350.     state.K = (dev->y_pixels_per_inch < 100 ? 2 : 4);
  351.     state.EndOfLine = true;
  352.     state.EncodedByteAlign = true;
  353.     dir = dir_mono_template;
  354.     dir.Compression.value = Compression_CCITT_T4;
  355.     dir.T4T6Options.tag = TIFFTAG_T4Options;
  356.     dir.T4T6Options.value = T4Options_2D_encoding | T4Options_fill_bits;
  357.     return tifff_print_page(dev, prn_stream, &state, &dir);
  358. }
  359. private int
  360. tiffg4_print_page(gx_device_printer *dev, FILE *prn_stream)
  361. {    stream_CFE_state state;
  362.     tiff_mono_directory dir;
  363.  
  364.     gdev_fax_init_state(&state, dev);
  365.     state.K = -1;
  366.     /*state.EncodedByteAlign = false;*/  /* no fill_bits option for T6 */
  367.     dir = dir_mono_template;
  368.     dir.Compression.value = Compression_CCITT_T6;
  369.     dir.T4T6Options.tag = TIFFTAG_T6Options;
  370.     return tifff_print_page(dev, prn_stream, &state, &dir);
  371. }
  372.  
  373. /* Print an LZW page. */
  374. private int
  375. tifflzw_print_page(gx_device_printer *dev, FILE *prn_stream)
  376. {    tiff_mono_directory dir;
  377.     stream_LZW_state state;
  378.     int code;
  379.  
  380.     dir = dir_mono_template;
  381.     dir.Compression.value = Compression_LZW;
  382.     dir.FillOrder.value = FillOrder_MSB2LSB;
  383.     tfax_begin_page(tfdev, prn_stream, &dir);
  384.     state.InitialCodeLength = 8;
  385.     state.FirstBitLowOrder = false;
  386.     state.BlockData = false;
  387.     state.EarlyChange = 0;        /****** CHECK THIS ******/
  388.     code = gdev_stream_print_page(dev, prn_stream, &s_LZWE_template,
  389.                       (stream_state *)&state);
  390.     gdev_tiff_end_page(&tfdev->tiff, prn_stream);
  391.     return code;
  392. }
  393.  
  394. /* Print a PackBits page. */
  395. private int
  396. tiffpack_print_page(gx_device_printer *dev, FILE *prn_stream)
  397. {    tiff_mono_directory dir;
  398.     stream_RLE_state state;
  399.     int code;
  400.  
  401.     dir = dir_mono_template;
  402.     dir.Compression.value = Compression_PackBits;
  403.     dir.FillOrder.value = FillOrder_MSB2LSB;
  404.     tfax_begin_page(tfdev, prn_stream, &dir);
  405.     state.EndOfData = false;
  406.     state.record_size = gdev_mem_bytes_per_scan_line((gx_device *)dev);
  407.     code = gdev_stream_print_page(dev, prn_stream, &s_RLE_template,
  408.                       (stream_state *)&state);
  409.     gdev_tiff_end_page(&tfdev->tiff, prn_stream);
  410.     return code;
  411. }
  412.  
  413. #undef tfdev
  414.  
  415. /* Begin a TIFF fax page. */
  416. private int
  417. tfax_begin_page(gx_device_tfax *tfdev, FILE *fp,
  418.   const tiff_mono_directory *pdir)
  419. {    return gdev_tiff_begin_page((gx_device_printer *)tfdev,
  420.                 &tfdev->tiff, fp,
  421.                 (const TIFF_dir_entry *)pdir,
  422.                 sizeof(*pdir) / sizeof(TIFF_dir_entry),
  423.                 NULL, 0);
  424. }
  425.